<?php
declare (strict_types = 1);

namespace app\admin;

use think\App;
use think\exception\ValidateException;
use think\Validate;

/**
 * 控制器基础类
 */
abstract class BaseController
{
    /**
     * Request实例
     * @var \think\Request
     */
    protected $request;

    /**
     * 应用实例
     * @var \think\App
     */
    protected $app;

    /**
     * 是否批量验证
     * @var bool
     */
    protected $batchValidate = false;

    /**
     * 控制器中间件
     * @var array
     */
    protected $middleware = [];

    /**
     * 构造方法
     * @access public
     * @param  App  $app  应用对象
     */
    public function __construct(App $app)
    {
        $this->app     = $app;
        $this->request = $this->app->request;

        // 控制器初始化
        $this->initialize();
    }

    // 初始化
    protected function initialize()
    {}

    /**
     * 验证数据
     * @access protected
     * @param  array        $data     数据
     * @param  string|array $validate 验证器名或者验证规则数组
     * @param  array        $message  提示信息
     * @param  bool         $batch    是否批量验证
     * @return array|string|true
     * @throws ValidateException
     */
    protected function validate(array $data, $validate, array $message = [], bool $batch = false)
    {
        if (is_array($validate)) {
            $v = new Validate();
            $v->rule($validate);
        } else {
            if (strpos($validate, '.')) {
                // 支持场景
                [$validate, $scene] = explode('.', $validate);
            }
            $class = false !== strpos($validate, '\\') ? $validate : $this->app->parseClass('validate', $validate);
            $v     = new $class();
            if (!empty($scene)) {
                $v->scene($scene);
            }
        }

        $v->message($message);

        // 是否批量验证
        if ($batch || $this->batchValidate) {
            $v->batch(true);
        }

        return $v->failException(true)->check($data);
    }
    /**
     * 断点下载方法
     * 
    */
    function dl_file_resume($file){
        //检测文件是否存在
        if (!is_file($file)){
            exit( json(['code'=>404,'msg'=>'未发现文件！']));
        }            
        $len = filesize($file);//获取文件大小
        $filename = basename($file);//获取文件名字
        $file_extension = strtolower(substr(strrchr($filename,"."),1));//获取文件扩展名
        //根据扩展名 指出输出浏览器格式
        switch( $file_extension ) {
            case "exe": $ctype="application/octet-stream"; break;
            case "zip": $ctype="application/zip"; break;
            case "mp3": $ctype="audio/mpeg"; break;
            case "mpg":$ctype="video/mpeg"; break;
            case "avi": $ctype="video/x-msvideo"; break;
            case "pdf": $ctype="application/pdf"; break;
            case "jpg": $ctype="image/jpeg"; break;
            case "jpeg": $ctype="image/jpeg"; break;
            case "png": $ctype="image/png"; break;
            case "webp": $ctype="image/webp"; break;
            case "xls": $ctype="application/vnd.ms-excel"; break;
            case "xlsx": $ctype="application/vnd.openxmlformats-officedocument.spreadsheetml.sheet"; break;
            case "doc": $ctype="application/msword"; break;
            case "docx": $ctype="application/vnd.openxmlformats-officedocument.wordprocessingml.document"; break;
            case "ppt": $ctype="application/vnd.ms-powerpoint"; break;
            case "pptx": $ctype="application/vnd.openxmlformats-officedocument.presentationml.presentation"; break;
            case "mp4": $ctype="video/mp4"; break;
            case "mpga": $ctype="audio/mpeg"; break;
    
            default: $ctype="application/force-download";
    
        }
        //Begin writing headers
        header("Cache-Control:");
        header("Cache-Control: public");

        //设置输出浏览器格式
        header("Content-Type: $ctype");
    
        if (strstr($_SERVER['HTTP_USER_AGENT'], "MSIE")) {//如果是IE浏览器
            # workaround for IE filename bug with multiple periods / multiple dots in filename
            # that adds square brackets to filename - eg. setup.abc.exe becomes setup[1].abc.exe
            $iefilename = preg_replace('/\./', '%2e', $filename, substr_count($filename, '.') - 1);
            header("Content-Disposition: attachment; filename=\"$iefilename\"");
        } else {
            header("Content-Disposition: attachment; filename=\"$filename\"");
        }
        header("Accept-Ranges: bytes");
        $size=filesize($file);
        //如果有$_SERVER['HTTP_RANGE']参数
        if(isset($_SERVER['HTTP_RANGE'])) {
    /*   ---------------------------
    
       Range头域 　　
       Range头域可以请求实体的一个或者多个子范围。
       例如， 　　
       表示头500个字节：bytes=0-499 　　表示第二个500字节：
       bytes=500-999 　　表示最后500个字节：bytes=-500 　　表示500字节以后的范围：
       bytes=500- 　　第一个和最后一个字节：bytes=0-0,-1 　　同时指定几个范围：
       bytes=500-600,601-999 　　但是服务器可以忽略此请求头，如果无条件GET包含Range请求头，响应会以状态码206（PartialContent）返回而不是以200 （OK）。
       ---------------------------*/
    // 断点后再次连接 $_SERVER['HTTP_RANGE'] 的值 bytes=4390912-
            list($a, $range)=explode("=",$_SERVER['HTTP_RANGE']);
       //if yes, download missing part
            str_replace($range, "-", $range);//这句干什么的呢。。。。
            $size2=$size-1;//文件总字节数
            $new_length=$size2-$range;//获取下次下载的长度
            header("HTTP/1.1 206 Partial Content");
            header("Content-Length: $new_length");//输入总长
            header("Content-Range: bytes $range$size2/$size");//Content-Range: bytes 4908618-4988927/4988928   95%的时候
        } else {//第一次连接
            $size2=$size-1;
            header("Content-Range: bytes 0-$size2/$size"); //Content-Range: bytes 0-4988927/4988928
            header("Content-Length: ".$size);//输出总长
        }
        //打开文件
        $fp=fopen("$file","rb");
        //设置指针位置
        fseek($fp,$range);
        //虚幻输出
        while(!feof($fp)){
            //设置文件最长执行时间
            set_time_limit(0);
            print(fread($fp,1024*8));//输出文件
            flush();//输出缓冲
            ob_flush();
        }
        fclose($fp);
        exit(json(['code'=>200,'msg'=>'加载完成下载内容']));
    }
}
